Entity relationship diagrams

The “crow’s foot” that connects the tables together shows us how the columns in one table relate to the columns in another table. ## Types Of Statements A few statements include: * CREATE TABLE is a statement that creates a new table in a database. * DROP TABLE is a statement that removes a table in a database. * SELECT allows you to read data and display it. This is called a query.

SELECT & FROM Statements

Here you were introduced to two statements that will be used in every query you write * SELECT - what columns you want back. * FROM - what table you are querying from.

SELECT id, account_id, occurred_at
FROM orders;

LIMIT Statement

The LIMIT statement is useful when you want to see just the first few rows of a table. This can be much faster for loading than if we load the entire dataset. The LIMIT command is always the very last part of a query. An example of showing just the first 10 rows of the orders table with all of the columns might look like the following:

SELECT *
FROM orders
LIMIT 10

ORDER BY Statement

SELECT id, occurred_at, total_amt_usd
FROM orders
ORDER BY occurred_at
LIMIT 10;

We can ORDER BY more than one column at a time. The statement sorts according to columns listed from left first and those listed on the right after that.

SELECT *
FROM orders
ORDER BY occurred_at, total_amt_usd
LIMIT 10;

WHERE Statements

Using the WHERE statement, we can subset out tables based on conditions that must be met.

SELECT *
FROM orders
WHERE gloss_amt_usd >= 1000
LIMIT 5;

The WHERE statement can also be used with non-numerical data. We can use the = and != operators here. You also need to be sure to use single quotes

SELECT name, website, primary_poc
FROM accounts
WHERE name = 'Exxon Mobil';

Arithmetic Operators

Derived Columns

Creating a new column that is a combination of existing columns is known as a derived column. ### Order of Operations Remember BOMDAS from maths class? The same order of operations apply when using arithmetic operators in SQL.

SELECT id, account_id, 
       poster_amt_usd/(standard_amt_usd + gloss_amt_usd + poster_amt_usd) AS post_per
FROM orders;

Introduction to Logical Operators

LIKE - This allows you to perform operations similar to using WHERE and =, but for cases when you might not know exactly what you are looking for. The LIKE operator is frequently used with %. The % tells us that we might want any number of characters leading up to a particular set of characters or following a certain set of characters.

/* select all the companies whose names start with 'C' */
SELECT name 
FROM accounts
WHERE name like 'C%'

/* select All companies whose names contain the string 'one' somewhere in the name. */
SELECT name
FROM accounts
WHERE name LIKE '%one%';

IN - This allows you to perform operations similar to using WHERE and =, but for more than one condition. We can check one, two or many column values for which we want to pull data, but all within the same query. TIP - In most SQL environments, you can use single or double quotation marks - and you may NEED to use double quotation marks if you have an apostrophe within the text you are attempting to pull.

SELECT name, primary_poc, sales_rep_id
FROM accounts
WHERE name IN ('Walmart', 'Target', 'Nordstrom');

NOT - This is used with IN and LIKE to select all of the rows NOT LIKE or NOT IN a certain condition.

/* Use the accounts table to find the account name, primary poc, and sales rep id for all stores except Walmart, Target, and Nordstrom */
SELECT name, primary_poc, sales_rep_id
FROM accounts
WHERE name NOT IN ('Walmart', 'Target', 'Nordstrom');

AND & BETWEEN - These allow you to combine operations where all combined conditions must be true. * The AND operator is used within a WHERE statement to consider more than one logical clause at a time. Each time you link a new statement with an AND, you will need to specify the column you are interested in looking at. * Sometimes we can make a cleaner statement using BETWEEN than we can using AND. Particularly this is true when we are using the same column for different parts of our AND statement. e.g.:

/*Instead of writing : WHERE column >= 6 AND column <= 10 we can instead write*/ 
WHERE column BETWEEN 6 AND 10
/* Write a query that returns all the orders where the standard_qty is over 1000, the poster_qty is 0, and the gloss_qty is 0 */
SELECT *
FROM orders
WHERE standard_qty > 1000 AND poster_qty = 0 AND gloss_qty = 0;

/* Using the accounts table find all the companies whose names do not start with 'C' and end with 's' */
SELECT name
FROM accounts
WHERE name NOT LIKE 'C%' AND name LIKE '%s';

/* Use the web_events table to find all information regarding individuals who were contacted via organic or adwords and started their account at any point in 2016 sorted from newest to oldest */ 
SELECT *
FROM web_events
WHERE channel IN ('organic', 'adwords') AND occurred_at BETWEEN '2016-01-01' AND '2017-01-01'
ORDER BY occurred_at DESC;
/*  You will notice that using BETWEEN is tricky for dates! While BETWEEN is generally inclusive of endpoints, it assumes the time is at 00:00:00 (i.e. midnight) for dates. This is the reason why we set the right-side endpoint of the period at '2017-01-01'. */

OR - This allow you to combine operations where at least one of the combined conditions must be true. Similar to the AND operator, the OR operator can combine multiple statements. Each time you link a new statement with an OR, you will need to specify the column you are interested in looking at.

/* Find list of orders ids where either gloss_qty or poster_qty is greater than 4000. Only include the id field in the resulting table. */
SELECT id
FROM orders
WHERE gloss_qty > 4000 OR poster_qty > 4000;

/* Write a query that returns a list of orders where the standard_qty is zero and either the gloss_qty or poster_qty is over 1000. */
SELECT *
FROM orders
WHERE standard_qty = 0 AND (gloss_qty > 1000 OR poster_qty > 1000);

/* Find all the company names that start with a 'C' or 'W', and the primary contact contains 'ana' or 'Ana', but it doesn't contain 'eana'*/
SELECT *
FROM accounts
WHERE (name LIKE 'C%' OR name LIKE 'W%') 
           AND ((primary_poc LIKE '%ana%' OR primary_poc LIKE '%Ana%') 
           AND primary_poc NOT LIKE '%eana%');

Summary of basic comands:


JOINS

Inner joins

Only return rows that appear in both tables e.g. Using ERD: Pull standard_qty, gloss_qty, and poster_qty from the orders table, and the website and the primary_poc from the accounts table.

SELECT orders.standard_qty, orders.gloss_qty, 
       orders.poster_qty,  accounts.website, 
       accounts.primary_poc
FROM orders
JOIN accounts
ON orders.account_id = accounts.id

Notice that we need to specify every table a column comes from in the SELECT statement. ### ALIAS When we JOIN tables together, it is nice to give each table an alias. Frequently an alias is just the first letter of the table name.

FROM tablename t1 
JOIN tablename2 t2

Questions

  1. Provide a table for all web_events associated with account name of Walmart. There should be three columns. Be sure to include the primary_poc, time of the event, and the channel for each event. Additionally, you might choose to add a fourth column to assure only Walmart events were chosen.
SELECT a.primary_poc, w.occurred_at, w.channel, a.name
FROM web_events w
JOIN accounts a
ON w.account_id = a.id
WHERE a.name = 'Walmart';
  1. Provide a table that provides the region for each sales_rep along with their associated accounts. Your final table should include three columns: the region name, the sales rep name, and the account name. Sort the accounts alphabetically (A-Z) according to account name.
SELECT r.name region, s.name rep, a.name account
FROM sales_reps s
JOIN region r
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
ORDER BY a.name;
  1. Provide the name for each region for every order, as well as the account name and the unit price they paid (total_amt_usd/total) for the order. Your final table should have 3 columns: region name, account name, and unit price. A few accounts have 0 for total, so I divided by (total + 0.01) to assure not dividing by zero.
SELECT r.name region, a.name account, 
       o.total_amt_usd/(o.total + 0.01) unit_price
FROM region r
JOIN sales_reps s
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
JOIN orders o
ON o.account_id = a.id;

Outer joins

If we want to include data that only exists in one of the tables we ant to join we can use an outer join If there is not matching information in the JOINed table, then you will have columns with empty cells with a data type called NULL. Writing a LEFT join statement: Comparing LEFT and RIGHT join:

LS0tCnRpdGxlOiAiTm90ZXMgb24gU1FMIgpvdXRwdXQ6IGh0bWxfbm90ZWJvb2sKLS0tCiMjIEVudGl0eSByZWxhdGlvbnNoaXAgZGlhZ3JhbXMKVGhlICJjcm93J3MgZm9vdCIgdGhhdCBjb25uZWN0cyB0aGUgdGFibGVzIHRvZ2V0aGVyIHNob3dzIHVzIGhvdyB0aGUgY29sdW1ucyBpbiBvbmUgdGFibGUgcmVsYXRlIHRvIHRoZSBjb2x1bW5zIGluIGFub3RoZXIgdGFibGUuIAohW10oaW1hZ2VzL0VSRC5wbmcpCiMjIFR5cGVzIE9mIFN0YXRlbWVudHMKIEEgZmV3IHN0YXRlbWVudHMgaW5jbHVkZToKKiBDUkVBVEUgVEFCTEUgaXMgYSBzdGF0ZW1lbnQgdGhhdCBjcmVhdGVzIGEgbmV3IHRhYmxlIGluIGEgZGF0YWJhc2UuCiogRFJPUCBUQUJMRSBpcyBhIHN0YXRlbWVudCB0aGF0IHJlbW92ZXMgYSB0YWJsZSBpbiBhIGRhdGFiYXNlLgoqIFNFTEVDVCBhbGxvd3MgeW91IHRvIHJlYWQgZGF0YSBhbmQgZGlzcGxheSBpdC4gVGhpcyBpcyBjYWxsZWQgYSBxdWVyeS4KCiMjIFNFTEVDVCAmIEZST00gU3RhdGVtZW50cwpIZXJlIHlvdSB3ZXJlIGludHJvZHVjZWQgdG8gdHdvIHN0YXRlbWVudHMgdGhhdCB3aWxsIGJlIHVzZWQgaW4gZXZlcnkgcXVlcnkgeW91IHdyaXRlCiogU0VMRUNUIC0gd2hhdCBjb2x1bW5zIHlvdSB3YW50IGJhY2suCiogRlJPTSAtIHdoYXQgdGFibGUgeW91IGFyZSBxdWVyeWluZyBmcm9tLiAKYGBge3NxbH0KU0VMRUNUIGlkLCBhY2NvdW50X2lkLCBvY2N1cnJlZF9hdApGUk9NIG9yZGVyczsKYGBgCgojIyBMSU1JVCBTdGF0ZW1lbnQKIFRoZSBMSU1JVCBzdGF0ZW1lbnQgaXMgdXNlZnVsIHdoZW4geW91IHdhbnQgdG8gc2VlIGp1c3QgdGhlIGZpcnN0IGZldyByb3dzIG9mIGEgdGFibGUuIFRoaXMgY2FuIGJlIG11Y2ggZmFzdGVyIGZvciBsb2FkaW5nIHRoYW4gaWYgd2UgbG9hZCB0aGUgZW50aXJlIGRhdGFzZXQuClRoZSBMSU1JVCBjb21tYW5kIGlzIGFsd2F5cyB0aGUgdmVyeSBsYXN0IHBhcnQgb2YgYSBxdWVyeS4gQW4gZXhhbXBsZSBvZiBzaG93aW5nIGp1c3QgdGhlIGZpcnN0IDEwIHJvd3Mgb2YgdGhlIG9yZGVycyB0YWJsZSB3aXRoIGFsbCBvZiB0aGUgY29sdW1ucyBtaWdodCBsb29rIGxpa2UgdGhlIGZvbGxvd2luZzoKYGBge3NxbH0KU0VMRUNUICoKRlJPTSBvcmRlcnMKTElNSVQgMTAKYGBgCgojIyBPUkRFUiBCWSBTdGF0ZW1lbnQKKiBUaGUgT1JERVIgQlkgc3RhdGVtZW50IGFsbG93cyB1cyB0byBvcmRlciBvdXIgdGFibGUgYnkgYW55IHJvdy4gCiogUmVtZW1iZXIgREVTQyBjYW4gYmUgYWRkZWQgYWZ0ZXIgdGhlIGNvbHVtbiBpbiB5b3VyIE9SREVSIEJZIHN0YXRlbWVudCB0byBzb3J0IGluIGRlc2NlbmRpbmcgb3JkZXIsIGFzIHRoZSBkZWZhdWx0IGlzIHRvIHNvcnQgaW4gYXNjZW5kaW5nIG9yZGVyLgpgYGB7c3FsfQpTRUxFQ1QgaWQsIG9jY3VycmVkX2F0LCB0b3RhbF9hbXRfdXNkCkZST00gb3JkZXJzCk9SREVSIEJZIG9jY3VycmVkX2F0CkxJTUlUIDEwOwpgYGAKCldlIGNhbiBPUkRFUiBCWSBtb3JlIHRoYW4gb25lIGNvbHVtbiBhdCBhIHRpbWUuIFRoZSBzdGF0ZW1lbnQgc29ydHMgYWNjb3JkaW5nIHRvIGNvbHVtbnMgbGlzdGVkIGZyb20gbGVmdCBmaXJzdCBhbmQgdGhvc2UgbGlzdGVkIG9uIHRoZSByaWdodCBhZnRlciB0aGF0LgpgYGB7c3FsfQpTRUxFQ1QgKgpGUk9NIG9yZGVycwpPUkRFUiBCWSBvY2N1cnJlZF9hdCwgdG90YWxfYW10X3VzZApMSU1JVCAxMDsKYGBgCgojIyBXSEVSRSBTdGF0ZW1lbnRzClVzaW5nIHRoZSAqKldIRVJFKiogc3RhdGVtZW50LCB3ZSBjYW4gc3Vic2V0IG91dCB0YWJsZXMgYmFzZWQgb24gY29uZGl0aW9ucyB0aGF0IG11c3QgYmUgbWV0LgpgYGB7c3FsfQpTRUxFQ1QgKgpGUk9NIG9yZGVycwpXSEVSRSBnbG9zc19hbXRfdXNkID49IDEwMDAKTElNSVQgNTsKYGBgCgpUaGUgV0hFUkUgc3RhdGVtZW50IGNhbiBhbHNvIGJlIHVzZWQgd2l0aCBub24tbnVtZXJpY2FsIGRhdGEuIFdlIGNhbiB1c2UgdGhlID0gYW5kICE9IG9wZXJhdG9ycyBoZXJlLiBZb3UgYWxzbyBuZWVkIHRvIGJlIHN1cmUgdG8gdXNlIHNpbmdsZSBxdW90ZXMKYGBge3NxbH0KU0VMRUNUIG5hbWUsIHdlYnNpdGUsIHByaW1hcnlfcG9jCkZST00gYWNjb3VudHMKV0hFUkUgbmFtZSA9ICdFeHhvbiBNb2JpbCc7CmBgYAoKIyMgQXJpdGhtZXRpYyBPcGVyYXRvcnMKIyMjIERlcml2ZWQgQ29sdW1ucwpDcmVhdGluZyBhIG5ldyBjb2x1bW4gdGhhdCBpcyBhIGNvbWJpbmF0aW9uIG9mIGV4aXN0aW5nIGNvbHVtbnMgaXMga25vd24gYXMgYSBkZXJpdmVkIGNvbHVtbi4KIyMjIE9yZGVyIG9mIE9wZXJhdGlvbnMKUmVtZW1iZXIgQk9NREFTIGZyb20gbWF0aHMgY2xhc3M/IFRoZSBzYW1lIG9yZGVyIG9mIG9wZXJhdGlvbnMgYXBwbHkgd2hlbiB1c2luZyBhcml0aG1ldGljIG9wZXJhdG9ycyBpbiBTUUwuCmBgYHtzcWx9ClNFTEVDVCBpZCwgYWNjb3VudF9pZCwgCiAgICAgICBwb3N0ZXJfYW10X3VzZC8oc3RhbmRhcmRfYW10X3VzZCArIGdsb3NzX2FtdF91c2QgKyBwb3N0ZXJfYW10X3VzZCkgQVMgcG9zdF9wZXIKRlJPTSBvcmRlcnM7CmBgYAojIyMgSW50cm9kdWN0aW9uIHRvIExvZ2ljYWwgT3BlcmF0b3JzCgoqKkxJS0UqKiAtIFRoaXMgYWxsb3dzIHlvdSB0byBwZXJmb3JtIG9wZXJhdGlvbnMgc2ltaWxhciB0byB1c2luZyBXSEVSRSBhbmQgPSwgYnV0IGZvciBjYXNlcyB3aGVuIHlvdSBtaWdodCBub3Qga25vdyBleGFjdGx5IHdoYXQgeW91IGFyZSBsb29raW5nIGZvci4gIFRoZSBMSUtFIG9wZXJhdG9yIGlzIGZyZXF1ZW50bHkgdXNlZCB3aXRoICUuIFRoZSAlIHRlbGxzIHVzIHRoYXQgd2UgbWlnaHQgd2FudCBhbnkgbnVtYmVyIG9mIGNoYXJhY3RlcnMgbGVhZGluZyB1cCB0byBhIHBhcnRpY3VsYXIgc2V0IG9mIGNoYXJhY3RlcnMgb3IgZm9sbG93aW5nIGEgY2VydGFpbiBzZXQgb2YgY2hhcmFjdGVycy4KYGBge3NxbH0KLyogc2VsZWN0IGFsbCB0aGUgY29tcGFuaWVzIHdob3NlIG5hbWVzIHN0YXJ0IHdpdGggJ0MnICovClNFTEVDVCBuYW1lIApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgbGlrZSAnQyUnCgovKiBzZWxlY3QgQWxsIGNvbXBhbmllcyB3aG9zZSBuYW1lcyBjb250YWluIHRoZSBzdHJpbmcgJ29uZScgc29tZXdoZXJlIGluIHRoZSBuYW1lLiAqLwpTRUxFQ1QgbmFtZQpGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgTElLRSAnJW9uZSUnOwpgYGAKCioqSU4qKiAtIFRoaXMgYWxsb3dzIHlvdSB0byBwZXJmb3JtIG9wZXJhdGlvbnMgc2ltaWxhciB0byB1c2luZyBXSEVSRSBhbmQgPSwgYnV0IGZvciBtb3JlIHRoYW4gb25lIGNvbmRpdGlvbi4gIFdlIGNhbiBjaGVjayBvbmUsIHR3byBvciBtYW55IGNvbHVtbiB2YWx1ZXMgZm9yIHdoaWNoIHdlIHdhbnQgdG8gcHVsbCBkYXRhLCBidXQgYWxsIHdpdGhpbiB0aGUgc2FtZSBxdWVyeS4gClRJUCAtIEluIG1vc3QgU1FMIGVudmlyb25tZW50cywgeW91IGNhbiB1c2Ugc2luZ2xlIG9yIGRvdWJsZSBxdW90YXRpb24gbWFya3MgLSBhbmQgeW91IG1heSBORUVEIHRvIHVzZSBkb3VibGUgcXVvdGF0aW9uIG1hcmtzIGlmIHlvdSBoYXZlIGFuIGFwb3N0cm9waGUgd2l0aGluIHRoZSB0ZXh0IHlvdSBhcmUgYXR0ZW1wdGluZyB0byBwdWxsLgpgYGB7c3FsfQpTRUxFQ1QgbmFtZSwgcHJpbWFyeV9wb2MsIHNhbGVzX3JlcF9pZApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgSU4gKCdXYWxtYXJ0JywgJ1RhcmdldCcsICdOb3Jkc3Ryb20nKTsKYGBgCioqTk9UKiogLSBUaGlzIGlzIHVzZWQgd2l0aCBJTiBhbmQgTElLRSB0byBzZWxlY3QgYWxsIG9mIHRoZSByb3dzIE5PVCBMSUtFIG9yIE5PVCBJTiBhIGNlcnRhaW4gY29uZGl0aW9uLgpgYGB7c3FsfQovKiBVc2UgdGhlIGFjY291bnRzIHRhYmxlIHRvIGZpbmQgdGhlIGFjY291bnQgbmFtZSwgcHJpbWFyeSBwb2MsIGFuZCBzYWxlcyByZXAgaWQgZm9yIGFsbCBzdG9yZXMgZXhjZXB0IFdhbG1hcnQsIFRhcmdldCwgYW5kIE5vcmRzdHJvbSAqLwpTRUxFQ1QgbmFtZSwgcHJpbWFyeV9wb2MsIHNhbGVzX3JlcF9pZApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgTk9UIElOICgnV2FsbWFydCcsICdUYXJnZXQnLCAnTm9yZHN0cm9tJyk7CmBgYAoKKipBTkQqKiAmICoqQkVUV0VFTioqIC0gVGhlc2UgYWxsb3cgeW91IHRvIGNvbWJpbmUgb3BlcmF0aW9ucyB3aGVyZSBhbGwgY29tYmluZWQgY29uZGl0aW9ucyBtdXN0IGJlIHRydWUuCiogVGhlIEFORCBvcGVyYXRvciBpcyB1c2VkIHdpdGhpbiBhIFdIRVJFIHN0YXRlbWVudCB0byBjb25zaWRlciBtb3JlIHRoYW4gb25lIGxvZ2ljYWwgY2xhdXNlIGF0IGEgdGltZS4gRWFjaCB0aW1lIHlvdSBsaW5rIGEgbmV3IHN0YXRlbWVudCB3aXRoIGFuIEFORCwgeW91IHdpbGwgbmVlZCB0byBzcGVjaWZ5IHRoZSBjb2x1bW4geW91IGFyZSBpbnRlcmVzdGVkIGluIGxvb2tpbmcgYXQuCiogU29tZXRpbWVzIHdlIGNhbiBtYWtlIGEgY2xlYW5lciBzdGF0ZW1lbnQgdXNpbmcgQkVUV0VFTiB0aGFuIHdlIGNhbiB1c2luZyBBTkQuIFBhcnRpY3VsYXJseSB0aGlzIGlzIHRydWUgd2hlbiB3ZSBhcmUgdXNpbmcgdGhlIHNhbWUgY29sdW1uIGZvciBkaWZmZXJlbnQgcGFydHMgb2Ygb3VyIEFORCBzdGF0ZW1lbnQuIGUuZy46CmBgYHtzcWx9Ci8qSW5zdGVhZCBvZiB3cml0aW5nIDogV0hFUkUgY29sdW1uID49IDYgQU5EIGNvbHVtbiA8PSAxMCB3ZSBjYW4gaW5zdGVhZCB3cml0ZSovIApXSEVSRSBjb2x1bW4gQkVUV0VFTiA2IEFORCAxMApgYGAKCmBgYHtzcWx9Ci8qIFdyaXRlIGEgcXVlcnkgdGhhdCByZXR1cm5zIGFsbCB0aGUgb3JkZXJzIHdoZXJlIHRoZSBzdGFuZGFyZF9xdHkgaXMgb3ZlciAxMDAwLCB0aGUgcG9zdGVyX3F0eSBpcyAwLCBhbmQgdGhlIGdsb3NzX3F0eSBpcyAwICovClNFTEVDVCAqCkZST00gb3JkZXJzCldIRVJFIHN0YW5kYXJkX3F0eSA+IDEwMDAgQU5EIHBvc3Rlcl9xdHkgPSAwIEFORCBnbG9zc19xdHkgPSAwOwoKLyogVXNpbmcgdGhlIGFjY291bnRzIHRhYmxlIGZpbmQgYWxsIHRoZSBjb21wYW5pZXMgd2hvc2UgbmFtZXMgZG8gbm90IHN0YXJ0IHdpdGggJ0MnIGFuZCBlbmQgd2l0aCAncycgKi8KU0VMRUNUIG5hbWUKRlJPTSBhY2NvdW50cwpXSEVSRSBuYW1lIE5PVCBMSUtFICdDJScgQU5EIG5hbWUgTElLRSAnJXMnOwoKLyogVXNlIHRoZSB3ZWJfZXZlbnRzIHRhYmxlIHRvIGZpbmQgYWxsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBpbmRpdmlkdWFscyB3aG8gd2VyZSBjb250YWN0ZWQgdmlhIG9yZ2FuaWMgb3IgYWR3b3JkcyBhbmQgc3RhcnRlZCB0aGVpciBhY2NvdW50IGF0IGFueSBwb2ludCBpbiAyMDE2IHNvcnRlZCBmcm9tIG5ld2VzdCB0byBvbGRlc3QgKi8gClNFTEVDVCAqCkZST00gd2ViX2V2ZW50cwpXSEVSRSBjaGFubmVsIElOICgnb3JnYW5pYycsICdhZHdvcmRzJykgQU5EIG9jY3VycmVkX2F0IEJFVFdFRU4gJzIwMTYtMDEtMDEnIEFORCAnMjAxNy0wMS0wMScKT1JERVIgQlkgb2NjdXJyZWRfYXQgREVTQzsKLyogIFlvdSB3aWxsIG5vdGljZSB0aGF0IHVzaW5nIEJFVFdFRU4gaXMgdHJpY2t5IGZvciBkYXRlcyEgV2hpbGUgQkVUV0VFTiBpcyBnZW5lcmFsbHkgaW5jbHVzaXZlIG9mIGVuZHBvaW50cywgaXQgYXNzdW1lcyB0aGUgdGltZSBpcyBhdCAwMDowMDowMCAoaS5lLiBtaWRuaWdodCkgZm9yIGRhdGVzLiBUaGlzIGlzIHRoZSByZWFzb24gd2h5IHdlIHNldCB0aGUgcmlnaHQtc2lkZSBlbmRwb2ludCBvZiB0aGUgcGVyaW9kIGF0ICcyMDE3LTAxLTAxJy4gKi8KYGBgCgoqKk9SKiogLSBUaGlzIGFsbG93IHlvdSB0byBjb21iaW5lIG9wZXJhdGlvbnMgd2hlcmUgYXQgbGVhc3Qgb25lIG9mIHRoZSBjb21iaW5lZCBjb25kaXRpb25zIG11c3QgYmUgdHJ1ZS4KU2ltaWxhciB0byB0aGUgQU5EIG9wZXJhdG9yLCB0aGUgT1Igb3BlcmF0b3IgY2FuIGNvbWJpbmUgbXVsdGlwbGUgc3RhdGVtZW50cy4gRWFjaCB0aW1lIHlvdSBsaW5rIGEgbmV3IHN0YXRlbWVudCB3aXRoIGFuIE9SLCB5b3Ugd2lsbCBuZWVkIHRvIHNwZWNpZnkgdGhlIGNvbHVtbiB5b3UgYXJlIGludGVyZXN0ZWQgaW4gbG9va2luZyBhdC4KYGBge3NxbH0KLyogRmluZCBsaXN0IG9mIG9yZGVycyBpZHMgd2hlcmUgZWl0aGVyIGdsb3NzX3F0eSBvciBwb3N0ZXJfcXR5IGlzIGdyZWF0ZXIgdGhhbiA0MDAwLiBPbmx5IGluY2x1ZGUgdGhlIGlkIGZpZWxkIGluIHRoZSByZXN1bHRpbmcgdGFibGUuICovClNFTEVDVCBpZApGUk9NIG9yZGVycwpXSEVSRSBnbG9zc19xdHkgPiA0MDAwIE9SIHBvc3Rlcl9xdHkgPiA0MDAwOwoKLyogV3JpdGUgYSBxdWVyeSB0aGF0IHJldHVybnMgYSBsaXN0IG9mIG9yZGVycyB3aGVyZSB0aGUgc3RhbmRhcmRfcXR5IGlzIHplcm8gYW5kIGVpdGhlciB0aGUgZ2xvc3NfcXR5IG9yIHBvc3Rlcl9xdHkgaXMgb3ZlciAxMDAwLiAqLwpTRUxFQ1QgKgpGUk9NIG9yZGVycwpXSEVSRSBzdGFuZGFyZF9xdHkgPSAwIEFORCAoZ2xvc3NfcXR5ID4gMTAwMCBPUiBwb3N0ZXJfcXR5ID4gMTAwMCk7CgovKiBGaW5kIGFsbCB0aGUgY29tcGFueSBuYW1lcyB0aGF0IHN0YXJ0IHdpdGggYSAnQycgb3IgJ1cnLCBhbmQgdGhlIHByaW1hcnkgY29udGFjdCBjb250YWlucyAnYW5hJyBvciAnQW5hJywgYnV0IGl0IGRvZXNuJ3QgY29udGFpbiAnZWFuYScqLwpTRUxFQ1QgKgpGUk9NIGFjY291bnRzCldIRVJFIChuYW1lIExJS0UgJ0MlJyBPUiBuYW1lIExJS0UgJ1clJykgCiAgICAgICAgICAgQU5EICgocHJpbWFyeV9wb2MgTElLRSAnJWFuYSUnIE9SIHByaW1hcnlfcG9jIExJS0UgJyVBbmElJykgCiAgICAgICAgICAgQU5EIHByaW1hcnlfcG9jIE5PVCBMSUtFICclZWFuYSUnKTsKYGBgCgojIyBTdW1tYXJ5IG9mIGJhc2ljIGNvbWFuZHM6CiFbXShpbWFnZXMvc3VtbWFyeV9iYXNpY19jb21tYW5kcy5wbmcpCgoqKioKCiMgSk9JTlMKIyMgSW5uZXIgam9pbnMgCk9ubHkgcmV0dXJuIHJvd3MgdGhhdCBhcHBlYXIgaW4gKipib3RoIHRhYmxlcyoqCiFbXShpbWFnZXMvRVJELnBuZykgCmUuZy4gVXNpbmcgRVJEOiBQdWxsIHN0YW5kYXJkX3F0eSwgZ2xvc3NfcXR5LCBhbmQgcG9zdGVyX3F0eSBmcm9tIHRoZSBvcmRlcnMgdGFibGUsIGFuZCB0aGUgd2Vic2l0ZSBhbmQgdGhlIHByaW1hcnlfcG9jIGZyb20gdGhlIGFjY291bnRzIHRhYmxlLgpgYGB7c3FsfQpTRUxFQ1Qgb3JkZXJzLnN0YW5kYXJkX3F0eSwgb3JkZXJzLmdsb3NzX3F0eSwgCiAgICAgICBvcmRlcnMucG9zdGVyX3F0eSwgIGFjY291bnRzLndlYnNpdGUsIAogICAgICAgYWNjb3VudHMucHJpbWFyeV9wb2MKRlJPTSBvcmRlcnMKSk9JTiBhY2NvdW50cwpPTiBvcmRlcnMuYWNjb3VudF9pZCA9IGFjY291bnRzLmlkCmBgYApOb3RpY2UgdGhhdCB3ZSBuZWVkIHRvIHNwZWNpZnkgZXZlcnkgdGFibGUgYSBjb2x1bW4gY29tZXMgZnJvbSBpbiB0aGUgU0VMRUNUIHN0YXRlbWVudC4KIyMjIEFMSUFTCldoZW4gd2UgSk9JTiB0YWJsZXMgdG9nZXRoZXIsIGl0IGlzIG5pY2UgdG8gZ2l2ZSBlYWNoIHRhYmxlIGFuIGFsaWFzLiBGcmVxdWVudGx5IGFuIGFsaWFzIGlzIGp1c3QgdGhlIGZpcnN0IGxldHRlciBvZiB0aGUgdGFibGUgbmFtZS4gCmBgYHtzcWx9CkZST00gdGFibGVuYW1lIHQxIApKT0lOIHRhYmxlbmFtZTIgdDIKYGBgCgojIyBRdWVzdGlvbnMKMS4gUHJvdmlkZSBhIHRhYmxlIGZvciBhbGwgKndlYl9ldmVudHMqIGFzc29jaWF0ZWQgd2l0aCBhY2NvdW50IG5hbWUgb2YgKldhbG1hcnQqLiBUaGVyZSBzaG91bGQgYmUgdGhyZWUgY29sdW1ucy4gQmUgc3VyZSB0byBpbmNsdWRlIHRoZSBwcmltYXJ5X3BvYywgdGltZSBvZiB0aGUgZXZlbnQsIGFuZCB0aGUgY2hhbm5lbCBmb3IgZWFjaCBldmVudC4gQWRkaXRpb25hbGx5LCB5b3UgbWlnaHQgY2hvb3NlIHRvIGFkZCBhIGZvdXJ0aCBjb2x1bW4gdG8gYXNzdXJlIG9ubHkgV2FsbWFydCBldmVudHMgd2VyZSBjaG9zZW4uIApgYGB7c3FsfQpTRUxFQ1QgYS5wcmltYXJ5X3BvYywgdy5vY2N1cnJlZF9hdCwgdy5jaGFubmVsLCBhLm5hbWUKRlJPTSB3ZWJfZXZlbnRzIHcKSk9JTiBhY2NvdW50cyBhCk9OIHcuYWNjb3VudF9pZCA9IGEuaWQKV0hFUkUgYS5uYW1lID0gJ1dhbG1hcnQnOwpgYGAKCjIuIFByb3ZpZGUgYSB0YWJsZSB0aGF0IHByb3ZpZGVzIHRoZSAqcmVnaW9uKiBmb3IgZWFjaCAqc2FsZXNfcmVwKiBhbG9uZyB3aXRoIHRoZWlyIGFzc29jaWF0ZWQgKmFjY291bnRzKi4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaW5jbHVkZSB0aHJlZSBjb2x1bW5zOiB0aGUgcmVnaW9uICpuYW1lKiwgdGhlIHNhbGVzIHJlcCAqbmFtZSosIGFuZCB0aGUgYWNjb3VudCAqbmFtZSouIFNvcnQgdGhlIGFjY291bnRzIGFscGhhYmV0aWNhbGx5IChBLVopIGFjY29yZGluZyB0byBhY2NvdW50IG5hbWUuIApgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lIHJlZ2lvbiwgcy5uYW1lIHJlcCwgYS5uYW1lIGFjY291bnQKRlJPTSBzYWxlc19yZXBzIHMKSk9JTiByZWdpb24gcgpPTiBzLnJlZ2lvbl9pZCA9IHIuaWQKSk9JTiBhY2NvdW50cyBhCk9OIGEuc2FsZXNfcmVwX2lkID0gcy5pZApPUkRFUiBCWSBhLm5hbWU7CmBgYAoKMy4gUHJvdmlkZSB0aGUgKm5hbWUqIGZvciBlYWNoIHJlZ2lvbiBmb3IgZXZlcnkgKm9yZGVyKiwgYXMgd2VsbCBhcyB0aGUgYWNjb3VudCAqbmFtZSogYW5kIHRoZSAqdW5pdCBwcmljZSogdGhleSBwYWlkICh0b3RhbF9hbXRfdXNkL3RvdGFsKSBmb3IgdGhlIG9yZGVyLiBZb3VyIGZpbmFsIHRhYmxlIHNob3VsZCBoYXZlIDMgY29sdW1uczogKnJlZ2lvbiBuYW1lLCBhY2NvdW50IG5hbWUsIGFuZCB1bml0IHByaWNlKi4gQSBmZXcgYWNjb3VudHMgaGF2ZSAwIGZvciAqdG90YWwqLCBzbyBJIGRpdmlkZWQgYnkgKHRvdGFsICsgMC4wMSkgdG8gYXNzdXJlIG5vdCBkaXZpZGluZyBieSB6ZXJvLgpgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lIHJlZ2lvbiwgYS5uYW1lIGFjY291bnQsIAogICAgICAgby50b3RhbF9hbXRfdXNkLyhvLnRvdGFsICsgMC4wMSkgdW5pdF9wcmljZQpGUk9NIHJlZ2lvbiByCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMucmVnaW9uX2lkID0gci5pZApKT0lOIGFjY291bnRzIGEKT04gYS5zYWxlc19yZXBfaWQgPSBzLmlkCkpPSU4gb3JkZXJzIG8KT04gby5hY2NvdW50X2lkID0gYS5pZDsKYGBgCgojIyBPdXRlciBqb2lucwpJZiB3ZSB3YW50IHRvIGluY2x1ZGUgZGF0YSB0aGF0IG9ubHkgZXhpc3RzIGluIG9uZSBvZiB0aGUgdGFibGVzIHdlIGFudCB0byBqb2luIHdlIGNhbiB1c2UgYW4gb3V0ZXIgam9pbgohW10oaW1hZ2VzL3R5cGVzX29mX2pvaW5zLnBuZykgCklmIHRoZXJlIGlzIG5vdCBtYXRjaGluZyBpbmZvcm1hdGlvbiBpbiB0aGUgSk9JTmVkIHRhYmxlLCB0aGVuIHlvdSB3aWxsIGhhdmUgY29sdW1ucyB3aXRoIGVtcHR5IGNlbGxzIHdpdGggYSBkYXRhIHR5cGUgY2FsbGVkIE5VTEwuCldyaXRpbmcgYSBMRUZUIGpvaW4gc3RhdGVtZW50OgohW10oaW1hZ2VzL2xlZnRfam9pbi5wbmcpIAohW10oaW1hZ2VzL2xlZnRfam9pbl9wdDIucG5nKSAKQ29tcGFyaW5nIExFRlQgYW5kIFJJR0hUIGpvaW46CiFbXShpbWFnZXMvY29tcGFyZV9sZWZ0X2FuZF9yaWdodF9qb2luLnBuZykg